home *** CD-ROM | disk | FTP | other *** search
/ Chip 2007 January, February, March & April / Chip-Cover-CD-2007-02.iso / Pakiet internetowy / Przegladarki internetowe / Mozilla Seamonkey 1.0.5 pl / seamonkey-1.0.5.pl-PL.win32.installer.exe / VENKMAN.XPI / bin / chrome / venkman.jar / content / venkman / tree-utils.js < prev    next >
Encoding:
JavaScript  |  2004-04-18  |  43.1 KB  |  1,677 lines

  1. /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*-
  2.  *
  3.  * ***** BEGIN LICENSE BLOCK *****
  4.  * Version: MPL 1.1/GPL 2.0/LGPL 2.1
  5.  *
  6.  * The contents of this file are subject to the Mozilla Public License Version
  7.  * 1.1 (the "License"); you may not use this file except in compliance with
  8.  * the License. You may obtain a copy of the License at
  9.  * http://www.mozilla.org/MPL/
  10.  *
  11.  * Software distributed under the License is distributed on an "AS IS" basis,
  12.  * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
  13.  * for the specific language governing rights and limitations under the
  14.  * License.
  15.  *
  16.  * The Original Code is The JavaScript Debugger.
  17.  *
  18.  * The Initial Developer of the Original Code is
  19.  * Netscape Communications Corporation.
  20.  * Portions created by the Initial Developer are Copyright (C) 1998
  21.  * the Initial Developer. All Rights Reserved.
  22.  *
  23.  * Contributor(s):
  24.  *   Robert Ginda, <rginda@netscape.com>, original author
  25.  *
  26.  * Alternatively, the contents of this file may be used under the terms of
  27.  * either the GNU General Public License Version 2 or later (the "GPL"), or
  28.  * the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
  29.  * in which case the provisions of the GPL or the LGPL are applicable instead
  30.  * of those above. If you wish to allow use of your version of this file only
  31.  * under the terms of either the GPL or the LGPL, and not to allow others to
  32.  * use your version of this file under the terms of the MPL, indicate your
  33.  * decision by deleting the provisions above and replace them with the notice
  34.  * and other provisions required by the GPL or the LGPL. If you do not delete
  35.  * the provisions above, a recipient may use your version of this file under
  36.  * the terms of any one of the MPL, the GPL or the LGPL.
  37.  *
  38.  * ***** END LICENSE BLOCK ***** */
  39.  
  40. /*
  41.  * BasicOView provides functionality of tree whose elements have no children.
  42.  * Usage:
  43.  * var myTree = new BasicOView()
  44.  * myTree.setColumnNames (["col 1", "col 2"]);
  45.  * myTree.data = [["row 1, col 1", "row 1, col 2"],
  46.  *                    ["row 2, col 1", "row 2, col 2"]];
  47.  * { override get*Properties, etc, as suits your purpose. }
  48.  *
  49.  * treeBoxObject.view = myTree;
  50.  * 
  51.  * You'll need to make the appropriate myTree.tree.invalidate calls
  52.  * when myTree.data changes.
  53.  */
  54.  
  55. function BasicOView()
  56. {
  57.     this.tree = null;
  58. }
  59.  
  60. /* functions *you* should call to initialize and maintain the tree state */
  61.  
  62. /* scroll the line specified by |line| to the center of the tree */
  63. BasicOView.prototype.centerLine =
  64. function bov_ctrln (line)
  65. {
  66.     var first = this.tree.getFirstVisibleRow();
  67.     var last = this.tree.getLastVisibleRow();
  68.     this.scrollToRow(line - total / 2);
  69. }
  70.  
  71. /* call this to set the association between column names and data columns */
  72. BasicOView.prototype.setColumnNames =
  73. function bov_setcn (aryNames)
  74. {
  75.     this.columnNames = new Object();
  76.     for (var i = 0; i < aryNames.length; ++i)
  77.         this.columnNames[aryNames[i]] = i;
  78. }
  79.  
  80. /*
  81.  * scroll the source so |line| is at either the top, center, or bottom
  82.  * of the view, depending on the value of |align|.
  83.  *
  84.  * line is the one based target line.
  85.  * if align is negative, the line will be scrolled to the top, if align is
  86.  * zero the line will be centered, and if align is greater than 0 the line
  87.  * will be scrolled to the bottom.  0 is the default.
  88.  */
  89. BasicOView.prototype.scrollTo =
  90. function bov_scrollto (line, align)
  91. {
  92.     if (!this.tree)
  93.         return;
  94.     
  95.     var headerRows = 1;
  96.     
  97.     var first = this.tree.getFirstVisibleRow();
  98.     var last  = this.tree.getLastVisibleRow();
  99.     var viz   = last - first - headerRows; /* total number of visible rows */
  100.  
  101.     /* all rows are visible, nothing to scroll */
  102.     if (first == 0 && last > this.rowCount)
  103.         return;
  104.     
  105.     /* tree lines are 0 based, we accept one based lines, deal with it */
  106.     --line;
  107.  
  108.     /* safety clamp */
  109.     if (line < 0)
  110.         line = 0;
  111.     if (line > this.rowCount)
  112.         line = this.rowCount;    
  113.  
  114.     if (align < 0)
  115.     {
  116.         if (line > this.rowCount - viz) /* overscroll, can't put a row from */
  117.             line = this.rowCount - viz; /* last page at the top. */
  118.         this.tree.scrollToRow(line);
  119.     }
  120.     else if (align > 0)
  121.     {
  122.         if (line < viz) /* underscroll, can't put a row from the first page */
  123.             line = 0;   /* at the bottom. */
  124.         else
  125.             line = line - viz + headerRows;
  126.         
  127.         this.tree.scrollToRow(line);
  128.     }
  129.     else
  130.     {
  131.         var half_viz = viz / 2;
  132.         /* lines past this line can't be centered without causing the tree
  133.          * to show more rows than we have. */
  134.         var lastCenterable = this.rowCount - half_viz;
  135.         if (line > lastCenterable)
  136.             line = lastCenterable;
  137.         /* lines before this can't be centered without causing the tree
  138.          * to attempt to display negative rows. */
  139.         else if (line < half_viz)
  140.             line = half_viz;
  141.         this.tree.scrollToRow(line - half_viz);
  142.     }                    
  143. }       
  144.  
  145. BasicOView.prototype.__defineGetter__("selectedIndex", bov_getsel);
  146. function bov_getsel()
  147. {
  148.     if (!this.tree || this.tree.view.selection.getRangeCount() < 1)
  149.         return -1;
  150.  
  151.     var min = new Object();
  152.     this.tree.view.selection.getRangeAt(0, min, {});
  153.     return min.value;
  154. }
  155.  
  156. BasicOView.prototype.__defineSetter__("selectedIndex", bov_setsel);
  157. function bov_setsel(i)
  158. {
  159.     if (i == -1)
  160.         this.tree.view.selection.clearSelection();
  161.     else
  162.         this.tree.view.selection.timedSelect (i, 500);
  163.     return i;
  164. }
  165.  
  166. /*
  167.  * functions the tree will call to retrieve the list state (nsITreeView.)
  168.  */
  169.  
  170. BasicOView.prototype.rowCount = 0;
  171.  
  172. BasicOView.prototype.getCellProperties =
  173. function bov_cellprops (row, col, properties)
  174. {
  175. }
  176.  
  177. BasicOView.prototype.getColumnProperties =
  178. function bov_colprops (col, properties)
  179. {
  180. }
  181.  
  182. BasicOView.prototype.getRowProperties =
  183. function bov_rowprops (index, properties)
  184. {
  185. }
  186.  
  187. BasicOView.prototype.isContainer =
  188. function bov_isctr (index)
  189. {
  190.     return false;
  191. }
  192.  
  193. BasicOView.prototype.isContainerOpen =
  194. function bov_isctropen (index)
  195. {
  196.     return false;
  197. }
  198.  
  199. BasicOView.prototype.isContainerEmpty =
  200. function bov_isctrempt (index)
  201. {
  202.     return false;
  203. }
  204.  
  205. BasicOView.prototype.isSeparator =
  206. function bov_isseparator (index)
  207. {
  208.     return false;
  209. }
  210.  
  211. BasicOView.prototype.isSorted =
  212. function bov_issorted (index)
  213. {
  214.     return false;
  215. }
  216.  
  217. BasicOView.prototype.canDrop =
  218. function bov_drop (index, orientation)
  219. {
  220.     return false;
  221. }
  222.  
  223. BasicOView.prototype.drop =
  224. function bov_drop (index, orientation)
  225. {
  226.     return false;
  227. }
  228.  
  229. BasicOView.prototype.getParentIndex =
  230. function bov_getpi (index)
  231. {
  232.     if (index < 0)
  233.         return -1;
  234.  
  235.     return 0;
  236. }
  237.  
  238. BasicOView.prototype.hasNextSibling =
  239. function bov_hasnxtsib (rowIndex, afterIndex)
  240. {
  241.     return (afterIndex < (this.rowCount - 1));
  242. }
  243.  
  244. BasicOView.prototype.getLevel =
  245. function bov_getlvl (index)
  246. {
  247.     return 0;
  248. }
  249.  
  250. BasicOView.prototype.getImageSrc =
  251. function bov_getimgsrc (row, col)
  252. {
  253. }
  254.  
  255. BasicOView.prototype.getProgressMode =
  256. function bov_getprgmode (row, col)
  257. {
  258. }
  259.  
  260. BasicOView.prototype.getCellValue =
  261. function bov_getcellval (row, col)
  262. {
  263. }
  264.  
  265. BasicOView.prototype.getCellText =
  266. function bov_getcelltxt (row, col)
  267. {
  268.     if (!this.columnNames)
  269.         return "";
  270.     
  271.     var ary = col.id.match (/:(.*)/);
  272.     if (ary)
  273.         col = ary[1];
  274.  
  275.     var colName = this.columnNames[col];
  276.     
  277.     if (typeof colName == "undefined")
  278.         return "";
  279.     
  280.     return this.data[row][colName];
  281. }
  282.  
  283. BasicOView.prototype.setTree =
  284. function bov_seto (tree)
  285. {
  286.     this.tree = tree;
  287. }
  288.  
  289. BasicOView.prototype.toggleOpenState =
  290. function bov_toggleopen (index)
  291. {
  292. }
  293.  
  294. BasicOView.prototype.cycleHeader =
  295. function bov_cyclehdr (col)
  296. {
  297. }
  298.  
  299. BasicOView.prototype.selectionChanged =
  300. function bov_selchg ()
  301. {
  302. }
  303.  
  304. BasicOView.prototype.cycleCell =
  305. function bov_cyclecell (row, col)
  306. {
  307. }
  308.  
  309. BasicOView.prototype.isEditable =
  310. function bov_isedit (row, col)
  311. {
  312.     return false;
  313. }
  314.  
  315. BasicOView.prototype.setCellValue =
  316. function bov_setct (row, col, value)
  317. {
  318. }
  319.  
  320. BasicOView.prototype.setCellText =
  321. function bov_setct (row, col, value)
  322. {
  323. }
  324.  
  325. BasicOView.prototype.onRouteFocus =
  326. function bov_rfocus (event)
  327. {
  328.     if ("onFocus" in this)
  329.         this.onFocus(event);
  330. }
  331.  
  332. BasicOView.prototype.onRouteBlur =
  333. function bov_rblur (event)
  334. {
  335.     if ("onBlur" in this)
  336.         this.onBlur(event);
  337. }
  338.  
  339. BasicOView.prototype.onRouteDblClick =
  340. function bov_rdblclick (event)
  341. {
  342.     if (!("onRowCommand" in this) || event.target.localName != "treechildren")
  343.         return;
  344.  
  345.     var rowIndex = this.tree.view.selection.currentIndex;
  346.     if (rowIndex == -1 || rowIndex > this.rowCount)
  347.         return;
  348.     var rec = this.childData.locateChildByVisualRow(rowIndex);
  349.     if (!rec)
  350.     {
  351.         ASSERT (0, "bogus row index " + rowIndex);
  352.         return;
  353.     }
  354.  
  355.     this.onRowCommand(rec, event);
  356. }
  357.  
  358. BasicOView.prototype.onRouteKeyPress =
  359. function bov_rkeypress (event)
  360. {
  361.     var rec;
  362.     var rowIndex;
  363.     
  364.     if ("onRowCommand" in this && (event.keyCode == 13 || event.charCode == 32))
  365.     {
  366.         if (!this.selection)
  367.             return;
  368.         
  369.         rowIndex = this.tree.view.selection.currentIndex;
  370.         if (rowIndex == -1 || rowIndex > this.rowCount)
  371.             return;
  372.         rec = this.childData.locateChildByVisualRow(rowIndex);
  373.         if (!rec)
  374.         {
  375.             ASSERT (0, "bogus row index " + rowIndex);
  376.             return;
  377.         }
  378.  
  379.         this.onRowCommand(rec, event);
  380.     }
  381.     else if ("onKeyPress" in this)
  382.     {
  383.         rowIndex = this.tree.view.selection.currentIndex;
  384.         if (rowIndex != -1 && rowIndex < this.rowCount)
  385.         {
  386.             rec = this.childData.locateChildByVisualRow(rowIndex);
  387.             if (!rec)
  388.             {
  389.                 ASSERT (0, "bogus row index " + rowIndex);
  390.                 return;
  391.             }
  392.         }
  393.         else
  394.         {
  395.             rec = null;
  396.         }
  397.         
  398.         this.onKeyPress(rec, event);
  399.     }
  400. }
  401.  
  402. BasicOView.prototype.performAction =
  403. function bov_pact (action)
  404. {
  405. }
  406.  
  407. BasicOView.prototype.performActionOnRow =
  408. function bov_pactrow (action)
  409. {
  410. }
  411.  
  412. BasicOView.prototype.performActionOnCell =
  413. function bov_pactcell (action)
  414. {
  415. }
  416.  
  417. /*
  418.  * record for the XULTreeView.  these things take care of keeping the
  419.  * XULTreeView properly informed of changes in value and child count.  you 
  420.  * shouldn't have to maintain tree state at all.
  421.  *
  422.  * |share| should be an otherwise empty object to store cache data.
  423.  * you should use the same object as the |share| for the XULTreeView that you
  424.  * indend to contain these records.
  425.  *
  426.  */
  427. function XULTreeViewRecord(share)
  428. {
  429.     this._share = share;
  430.     this.visualFootprint = 1;
  431.     //this.childIndex = -1;
  432.     this.isHidden = true; /* records are considered hidden until they are
  433.                            * inserted into a live tree */
  434. }
  435.  
  436. XULTreeViewRecord.prototype.isContainerOpen = false;
  437.  
  438. /*
  439.  * walk the parent tree to find our tree container.  return null if there is
  440.  * none
  441.  */
  442. XULTreeViewRecord.prototype.findContainerTree =
  443. function xtvr_gettree ()
  444. {
  445.     if (!("parentRecord" in this))
  446.         return null;
  447.     var parent = this.parentRecord;
  448.     
  449.     while (parent)
  450.     {
  451.         if ("_treeView" in parent)
  452.             return parent._treeView;
  453.         if ("parentRecord" in parent)
  454.             parent = parent.parentRecord;
  455.         else
  456.             parent = null;
  457.     }
  458.  
  459.     return null;
  460. }
  461.  
  462. XULTreeViewRecord.prototype.__defineGetter__("childIndex", xtvr_getChildIndex);
  463. function xtvr_getChildIndex ()
  464. {
  465.     //dd ("getChildIndex {");
  466.     
  467.     if (!("parentRecord" in this))
  468.     {
  469.         delete this._childIndex;
  470.         //dd ("} -1");
  471.         return -1;
  472.     }
  473.     
  474.     if ("_childIndex" in this)
  475.     {
  476.         if ("childData" in this && this._childIndex in this.childData &&
  477.             this.childData[this._childIndex] == this)
  478.         {
  479.             //dd ("} " + this._childIndex);
  480.             return this._childIndex;
  481.         }
  482.     }
  483.  
  484.     var childData = this.parentRecord.childData;
  485.     var len = childData.length;
  486.     for (var i = 0; i < len; ++i)
  487.     {
  488.         if (childData[i] == this)
  489.         {
  490.             this._childIndex = i;
  491.             //dd ("} " + this._childIndex);
  492.             return i;
  493.         }
  494.     }
  495.     
  496.     delete this._childIndex;
  497.     //dd ("} -1");
  498.     return -1;
  499. }
  500.  
  501. XULTreeViewRecord.prototype.__defineSetter__("childIndex", xtvr_setChildIndex);
  502. function xtvr_setChildIndex ()
  503. {
  504.     dump ("xtvr: childIndex is read only, ignore attempt to write to it\n");
  505.     if (typeof getStackTrace == "function")
  506.         dump (getStackTrace());
  507. }
  508.  
  509. /* count the number of parents, not including the root node */
  510. XULTreeViewRecord.prototype.__defineGetter__("level", xtvr_getLevel);
  511. function xtvr_getLevel ()
  512. {
  513.     if (!("parentRecord" in this))
  514.         return -1;
  515.     
  516.     var rv = 0;
  517.     var parentRecord = this.parentRecord;
  518.     while ("parentRecord" in parentRecord &&
  519.            (parentRecord = parentRecord.parentRecord)) ++rv;
  520.     return rv;
  521. }
  522.  
  523. /*
  524.  * associates a property name on this record, with a column in the tree.  This
  525.  * method will set up a get/set pair for the property name you specify which
  526.  * will take care of updating the tree when the value changes.  DO NOT try
  527.  * to change your mind later.  Do not attach a different name to the same colID,
  528.  * and do no rename the colID.  You have been warned.
  529.  */
  530. XULTreeViewRecord.prototype.setColumnPropertyName =
  531. function xtvr_setcol (colID, propertyName)
  532. {
  533.     function xtvr_getValueShim ()
  534.     {
  535.         return this._colValues[colID];
  536.     }
  537.     function xtvr_setValueShim (newValue)
  538.     {
  539.         this._colValues[colID] = newValue;
  540.         /* XXX this.invalidate(); */
  541.         return newValue;
  542.     }
  543.  
  544.     if (!("_colValues" in this))
  545.         this._colValues = new Object();
  546.     
  547.     if (typeof propertyName == "function")
  548.     {
  549.         this._colValues.__defineGetter__(colID, propertyName);
  550.     }
  551.     else
  552.     {
  553.         this.__defineGetter__(propertyName, xtvr_getValueShim);
  554.         this.__defineSetter__(propertyName, xtvr_setValueShim);
  555.     }
  556. }
  557.  
  558. XULTreeViewRecord.prototype.setColumnPropertyValue =
  559. function xtvr_setcolv (colID, value)
  560. {
  561.     this._colValues[colID] = value;
  562. }
  563.  
  564. /*
  565.  * set the default sort column and resort.
  566.  */
  567. XULTreeViewRecord.prototype.setSortColumn =
  568. function xtvr_setcol (colID, dir)
  569. {
  570.     //dd ("setting sort column to " + colID);
  571.     this._share.sortColumn = colID;
  572.     this._share.sortDirection = (typeof dir == "undefined") ? 1 : dir;
  573.     this.resort();
  574. }
  575.  
  576. /*
  577.  * set the default sort direction.  1 is ascending, -1 is descending, 0 is no
  578.  * sort.  setting this to 0 will *not* recover the natural insertion order,
  579.  * it will only affect newly added items.
  580.  */
  581. XULTreeViewRecord.prototype.setSortDirection =
  582. function xtvr_setdir (dir)
  583. {
  584.     this._share.sortDirection = dir;
  585. }
  586.  
  587. /*
  588.  * invalidate this row in the tree
  589.  */
  590. XULTreeViewRecord.prototype.invalidate =
  591. function xtvr_invalidate()
  592. {
  593.     var tree = this.findContainerTree();
  594.     if (tree)
  595.     {
  596.         var row = this.calculateVisualRow();
  597.         if (row != -1)
  598.             tree.tree.invalidateRow(row);
  599.     }
  600. }
  601.  
  602. /*
  603.  * invalidate any data in the cache.
  604.  */
  605. XULTreeViewRecord.prototype.invalidateCache =
  606. function xtvr_killcache()
  607. {
  608.     this._share.rowCache = new Object();
  609.     this._share.lastComputedIndex = -1;
  610.     this._share.lastIndexOwner = null;
  611. }
  612.  
  613. /*
  614.  * default comparator function for sorts.  if you want a custom sort, override
  615.  * this method.  We declare xtvr_sortcmp as a top level function, instead of
  616.  * a function expression so we can refer to it later.
  617.  */
  618. XULTreeViewRecord.prototype.sortCompare = xtvr_sortcmp;
  619. function xtvr_sortcmp (a, b)
  620. {
  621.     var sc = a._share.sortColumn;
  622.     var sd = a._share.sortDirection;    
  623.     
  624.     a = a[sc];
  625.     b = b[sc];
  626.  
  627.     if (a < b)
  628.         return -1 * sd;
  629.     
  630.     if (a > b)
  631.         return 1 * sd;
  632.     
  633.     return 0;
  634. }
  635.  
  636. /*
  637.  * this method will cause all child records to be resorted.  any records
  638.  * with the default sortCompare method will be sorted by the colID passed to
  639.  * setSortColumn.
  640.  *
  641.  * the local parameter is used internally to control whether or not the 
  642.  * sorted rows are invalidated.  don't use it yourself.
  643.  */
  644. XULTreeViewRecord.prototype.resort =
  645. function xtvr_resort (leafSort)
  646. {
  647.     if (!("childData" in this) || this.childData.length < 1 ||
  648.         (this.childData[0].sortCompare == xtvr_sortcmp &&
  649.          !("sortColumn" in this._share) || this._share.sortDirection == 0))
  650.     {
  651.         /* if we have no children, or we have the default sort compare and no
  652.          * sort flags, then just exit */
  653.         return;
  654.     }
  655.  
  656.     this.childData.sort(this.childData[0].sortCompare);
  657.     
  658.     for (var i = 0; i < this.childData.length; ++i)
  659.     {
  660.         //this.childData[i].childIndex = i;
  661.         if ("isContainerOpen" in this.childData[i] &&
  662.             this.childData[i].isContainerOpen)
  663.             this.childData[i].resort(true);
  664.         else
  665.             this.childData[i].sortIsInvalid = true;
  666.     }
  667.     
  668.     if (!leafSort)
  669.     {
  670.         this.invalidateCache();
  671.         var tree = this.findContainerTree();
  672.         if (tree && tree.tree)
  673.         {
  674.             var rowIndex = this.calculateVisualRow();
  675.             /*
  676.             dd ("invalidating " + rowIndex + " - " +
  677.                 (rowIndex + this.visualFootprint - 1));
  678.             */
  679.             tree.tree.invalidateRange (rowIndex,
  680.                                        rowIndex + this.visualFootprint - 1);
  681.         }
  682.     }
  683.     /*
  684.     else
  685.         dd("not a leafSort");
  686.     */
  687.     delete this.sortIsInvalid;
  688. }
  689.     
  690. /*
  691.  * call this to indicate that this node may have children at one point.  make
  692.  * sure to call it before adding your first child.
  693.  */
  694. XULTreeViewRecord.prototype.reserveChildren =
  695. function xtvr_rkids (always)
  696. {
  697.     if (!("childData" in this))
  698.         this.childData = new Array();
  699.     if (!("isContainerOpen" in this))
  700.         this.isContainerOpen = false;
  701.     if (always)
  702.         this.alwaysHasChildren = true;
  703.     else
  704.         delete this.alwaysHasChildren;
  705. }
  706.  
  707. /*
  708.  * add a child to the end of the child list for this record.  takes care of 
  709.  * updating the tree as well.
  710.  */
  711. XULTreeViewRecord.prototype.appendChild =
  712. function xtvr_appchild (child)
  713. {
  714.     if (!(child instanceof XULTreeViewRecord))
  715.         throw Components.results.NS_ERROR_INVALID_ARG;
  716.     
  717.     child.isHidden = false;
  718.     child.parentRecord = this;
  719.     //child.childIndex = this.childData.length;
  720.     this.childData.push(child);
  721.     
  722.     if ("isContainerOpen" in this && this.isContainerOpen)
  723.     {
  724.         //dd ("appendChild: " + xtv_formatRecord(child, ""));
  725.         if (this.calculateVisualRow() >= 0)
  726.         {
  727.             var tree = this.findContainerTree();
  728.             if (tree && tree.frozen)
  729.                 this.needsResort = true;
  730.             else
  731.                 this.resort(true);  /* resort, don't invalidate.  we're going  
  732.                                      * to do that in the 
  733.                                      * onVisualFootprintChanged call. */
  734.         }
  735.         this.onVisualFootprintChanged(child.calculateVisualRow(),
  736.                                       child.visualFootprint);
  737.     }
  738. }
  739.  
  740. /*
  741.  * add a list of children to the end of the child list for this record.
  742.  * faster than multiple appendChild() calls.
  743.  */
  744. XULTreeViewRecord.prototype.appendChildren =
  745. function xtvr_appchild (children)
  746. {
  747.     var idx = this.childData.length;
  748.     var delta = 0;
  749.     var len = children.length;
  750.     for (var i = 0; i <  len; ++i)
  751.     {
  752.         var child = children[i];
  753.         child.isHidden = false;
  754.         child.parentRecord = this;
  755.         this.childData.push(child);
  756.         // this.childData[idx] = child;
  757.         //child.childIndex = idx++;
  758.         delta += child.visualFootprint;
  759.     }
  760.     
  761.     if ("isContainerOpen" in this && this.isContainerOpen)
  762.     {
  763.         if (this.calculateVisualRow() >= 0)
  764.         {
  765.             this.resort(true);  /* resort, don't invalidate.  we're going to do
  766.                                  * that in the onVisualFootprintChanged call. */
  767.         }
  768.         this.onVisualFootprintChanged(this.childData[0].calculateVisualRow(),
  769.                                       delta);
  770.     }
  771. }
  772.  
  773. /*
  774.  * remove a child from this record. updates the tree too.  DONT call this with
  775.  * an index not actually contained by this record.
  776.  */
  777. XULTreeViewRecord.prototype.removeChildAtIndex =
  778. function xtvr_remchild (index)
  779. {
  780.     if (!ASSERT(this.childData.length, "removing from empty childData"))
  781.         return;
  782.     
  783.     //for (var i = index + 1; i < this.childData.length; ++i)
  784.     //    --this.childData[i].childIndex;
  785.     
  786.     var orphan = this.childData[index];
  787.     var fpDelta = -orphan.visualFootprint;
  788.     var changeStart = orphan.calculateVisualRow();
  789.     //this.childData[index].childIndex = -1;
  790.     delete orphan.parentRecord;
  791.     arrayRemoveAt (this.childData, index);
  792.     
  793.     if (!orphan.isHidden && "isContainerOpen" in this && this.isContainerOpen)
  794.     {
  795.         this.onVisualFootprintChanged (changeStart, fpDelta);
  796.     }
  797. }
  798.  
  799. /*
  800.  * hide this record and all descendants.
  801.  */
  802. XULTreeViewRecord.prototype.hide =
  803. function xtvr_hide ()
  804. {
  805.     if (this.isHidden)
  806.         return;
  807.  
  808.     /* get the row before hiding */
  809.     var row = this.calculateVisualRow();
  810.     this.invalidateCache();
  811.     this.isHidden = true;
  812.     /* go right to the parent so we don't muck with our own visualFoorptint
  813.      * record, we'll need it to be correct if we're ever unHidden. */
  814.     if ("parentRecord" in this)
  815.         this.parentRecord.onVisualFootprintChanged (row, -this.visualFootprint);
  816. }
  817.  
  818. /*
  819.  * unhide this record and all descendants.
  820.  */
  821. XULTreeViewRecord.prototype.unHide =
  822. function xtvr_uhide ()
  823. {
  824.     if (!this.isHidden)
  825.         return;
  826.  
  827.     this.isHidden = false;
  828.     this.invalidateCache();
  829.     var row = this.calculateVisualRow();
  830.     if (this.parentRecord)
  831.         this.parentRecord.onVisualFootprintChanged (row, this.visualFootprint);
  832. }
  833.  
  834. /*
  835.  * open this record, exposing it's children.  DONT call this method if the
  836.  * record has no children.
  837.  */
  838. XULTreeViewRecord.prototype.open =
  839. function xtvr_open ()
  840. {
  841.     if (this.isContainerOpen)
  842.         return;
  843.  
  844.     if ("onPreOpen" in this)
  845.         this.onPreOpen();
  846.     
  847.     this.isContainerOpen = true;
  848.     var delta = 0;
  849.     for (var i = 0; i < this.childData.length; ++i)
  850.     {
  851.         if (!this.childData[i].isHidden)
  852.             delta += this.childData[i].visualFootprint;
  853.     }
  854.  
  855.     /* this resort should only happen if the sort column changed */
  856.     this.resort(true);
  857.     this.visualFootprint += delta;
  858.     if ("parentRecord" in this)
  859.     {
  860.         this.parentRecord.onVisualFootprintChanged(this.calculateVisualRow(),
  861.                                                    0);
  862.         this.parentRecord.onVisualFootprintChanged(this.calculateVisualRow() +
  863.                                                    1, delta);
  864.     }
  865. }
  866.  
  867. /*
  868.  * close this record, hiding it's children.  DONT call this method if the record
  869.  * has no children, or if it is already closed.
  870.  */
  871. XULTreeViewRecord.prototype.close =
  872. function xtvr_close ()
  873. {
  874.     if (!this.isContainerOpen)
  875.         return;
  876.     
  877.     this.isContainerOpen = false;
  878.     var delta = 1 - this.visualFootprint;
  879.     this.visualFootprint += delta;
  880.     if ("parentRecord" in this)
  881.     {
  882.         this.parentRecord.onVisualFootprintChanged(this.calculateVisualRow(),
  883.                                                    0);
  884.         this.parentRecord.onVisualFootprintChanged(this.calculateVisualRow() +
  885.                                                    1, delta);
  886.     }
  887.  
  888.     if ("onPostClose" in this)
  889.         this.onPostClose();
  890. }
  891.  
  892. /*
  893.  * called when a node above this one grows or shrinks.  we need to adjust
  894.  * our own visualFootprint to match the change, and pass the message on.
  895.  */
  896. XULTreeViewRecord.prototype.onVisualFootprintChanged =
  897. function xtvr_vpchange (start, amount)
  898. {
  899.     /* if we're not hidden, but this notification came from a hidden node 
  900.      * (start == -1), ignore it, it doesn't affect us. */
  901.     if (start == -1 && !this.isHidden)
  902.     {
  903.         
  904.         //dd ("vfp change (" + amount + ") from hidden node ignored.");
  905.         return;
  906.     }
  907.     
  908.     this.visualFootprint += amount;
  909.  
  910.     if ("parentRecord" in this)
  911.         this.parentRecord.onVisualFootprintChanged(start, amount);
  912. }
  913.  
  914. /*
  915.  * calculate the "visual" row for this record.  If the record isn't actually
  916.  * visible return -1.
  917.  * eg.
  918.  * Name        Visual Row
  919.  * node1       0
  920.  *   node11    1
  921.  *   node12    2
  922.  * node2       3
  923.  *   node21    4
  924.  * node3       5
  925.  */
  926. XULTreeViewRecord.prototype.calculateVisualRow =
  927. function xtvr_calcrow ()
  928. {
  929.     /* if this is the second time in a row that someone asked us, fetch the last
  930.      * result from the cache. */
  931.     if (this._share.lastIndexOwner == this)
  932.         return this._share.lastComputedIndex;
  933.  
  934.     var vrow;
  935.  
  936.         /* if this is an uninserted or hidden node, or... */
  937.     if (!("parentRecord" in this) || (this.isHidden) ||
  938.         /* if parent isn't open, or... */
  939.         (!this.parentRecord.isContainerOpen) ||
  940.         /* parent isn't visible */
  941.         ((vrow = this.parentRecord.calculateVisualRow()) == -1))
  942.     {
  943.         /* then we're not visible, return -1 */
  944.         //dd ("cvr: returning -1");
  945.         return -1;
  946.     }
  947.  
  948.     /* parent is the root node XXX parent is not visible */
  949.     if (vrow == null)
  950.         vrow = 0;
  951.     else
  952.     /* parent is not the root node, add one for the space they take up. */
  953.         ++vrow;
  954.  
  955.     /* add in the footprint for all of the earlier siblings */
  956.     var ci = this.childIndex;
  957.     for (var i = 0; i < ci; ++i)
  958.     {
  959.         if (!this.parentRecord.childData[i].isHidden)
  960.             vrow += this.parentRecord.childData[i].visualFootprint;
  961.     }
  962.  
  963.     /* save this calculation to the cache. */
  964.     this._share.lastIndexOwner = this;
  965.     this._share.lastComputedIndex = vrow;
  966.  
  967.     return vrow;
  968. }
  969.  
  970. /*
  971.  * locates the child record for the visible row |targetRow|.  DO NOT call this
  972.  * with a targetRow less than this record's visual row, or greater than this
  973.  * record's visual row + the number of visible children it has.
  974.  */
  975. XULTreeViewRecord.prototype.locateChildByVisualRow =
  976. function xtvr_find (targetRow, myRow)
  977. {
  978.     if (targetRow in this._share.rowCache)
  979.         return this._share.rowCache[targetRow];
  980.  
  981.     /* if this is true, we *are* the index */
  982.     if (targetRow == myRow)
  983.         return (this._share.rowCache[targetRow] = this);
  984.  
  985.     /* otherwise, we've got to search the kids */
  986.     var childStart = myRow; /* childStart represents the starting visual row
  987.                              * for the child we're examining. */
  988.     for (var i = 0; i < this.childData.length; ++i)
  989.     {
  990.         var child = this.childData[i];
  991.         /* ignore hidden children */
  992.         if (child.isHidden)
  993.             continue;
  994.         /* if this kid is the targetRow, we're done */
  995.         if (childStart == targetRow)
  996.             return (this._share.rowCache[targetRow] = child);
  997.         /* if this kid contains the index, ask *it* to find the record */
  998.         else if (targetRow <= childStart + child.visualFootprint) {
  999.             /* this *has* to succeed */
  1000.             var rv = child.locateChildByVisualRow(targetRow, childStart + 1);
  1001.             //XXXASSERT (rv, "Can't find a row that *has* to be there.");
  1002.             /* don't cache this, the previous call to locateChildByVisualRow
  1003.              * just did. */
  1004.             return rv;
  1005.         }
  1006.  
  1007.         /* otherwise, get ready to ask the next kid */
  1008.         childStart += child.visualFootprint;
  1009.     }
  1010.  
  1011.     return null;
  1012. }   
  1013.  
  1014. /* XTLabelRecords can be used to drop a label into an arbitrary place in an
  1015.  * arbitrary tree.  normally, specializations of XULTreeViewRecord are tied to
  1016.  * a specific tree because of implementation details.  XTLabelRecords are
  1017.  * specially designed (err, hacked) to work around these details.  this makes
  1018.  * them slower, but more generic.
  1019.  *
  1020.  * we set up a getter for _share that defers to the parent object.  this lets
  1021.  * XTLabelRecords work in any tree.
  1022.  */
  1023. function XTLabelRecord (columnName, label, blankCols)
  1024. {
  1025.     this.setColumnPropertyName (columnName, "label");
  1026.     this.label = label;
  1027.     this.property = null;
  1028.     
  1029.     if (typeof blankCols == "object")
  1030.     {
  1031.         for (var i in blankCols)
  1032.             this._colValues[blankCols[i]] = "";
  1033.     }
  1034. }
  1035.  
  1036. XTLabelRecord.prototype = new XULTreeViewRecord (null);
  1037.  
  1038. XTLabelRecord.prototype.__defineGetter__("_share", tolr_getshare);
  1039. function tolr_getshare()
  1040. {
  1041.     if ("parentRecord" in this)
  1042.         return this.parentRecord._share;
  1043.  
  1044.     ASSERT (0, "XTLabelRecord cannot be the root of a visible tree.");
  1045.     return null;
  1046. }
  1047.  
  1048. /* XTRootRecord is used internally by XULTreeView, you probably don't need to 
  1049.  * make any of these */ 
  1050. function XTRootRecord (tree, share)
  1051. {
  1052.     this._share = share;
  1053.     this._treeView = tree;
  1054.     this.visualFootprint = 0;
  1055.     this.isHidden = false;
  1056.     this.reserveChildren();
  1057.     this.isContainerOpen = true;
  1058. }
  1059.  
  1060. /* no cache passed in here, we set it in the XTRootRecord contructor instead. */
  1061. XTRootRecord.prototype = new XULTreeViewRecord (null);
  1062.  
  1063. XTRootRecord.prototype.open =
  1064. XTRootRecord.prototype.close =
  1065. function torr_notimplemented()
  1066. {
  1067.     /* don't do this on a root node */
  1068. }
  1069.  
  1070. XTRootRecord.prototype.calculateVisualRow =
  1071. function torr_calcrow ()
  1072. {
  1073.     return null;
  1074. }
  1075.  
  1076. XTRootRecord.prototype.resort =
  1077. function torr_resort ()
  1078. {
  1079.     if ("_treeView" in this && this._treeView.frozen)
  1080.     {
  1081.         this._treeView.needsResort = true;
  1082.         return;
  1083.     }
  1084.     
  1085.     if (!("childData" in this) || this.childData.length < 1 ||
  1086.         (this.childData[0].sortCompare == xtvr_sortcmp &&
  1087.          !("sortColumn" in this._share) || this._share.sortDirection == 0))
  1088.     {
  1089.         /* if we have no children, or we have the default sort compare but we're
  1090.          * missing a sort flag, then just exit */
  1091.         return;
  1092.     }
  1093.     
  1094.     this.childData.sort(this.childData[0].sortCompare);
  1095.     
  1096.     for (var i = 0; i < this.childData.length; ++i)
  1097.     {
  1098.         //this.childData[i].childIndex = i;
  1099.         if ("isContainerOpen" in this.childData[i] &&
  1100.             this.childData[i].isContainerOpen)
  1101.             this.childData[i].resort(true);
  1102.         else
  1103.             this.childData[i].sortIsInvalid = true;
  1104.     }
  1105.     
  1106.     if ("_treeView" in this && this._treeView.tree)
  1107.     {
  1108.         /*
  1109.         dd ("root node: invalidating 0 - " + this.visualFootprint +
  1110.             " for sort");
  1111.         */
  1112.         this.invalidateCache();
  1113.         this._treeView.tree.invalidateRange (0, this.visualFootprint);
  1114.     }
  1115. }
  1116.  
  1117. XTRootRecord.prototype.locateChildByVisualRow =
  1118. function torr_find (targetRow)
  1119. {
  1120.     if (targetRow in this._share.rowCache)
  1121.         return this._share.rowCache[targetRow];
  1122.  
  1123.     var childStart = -1; /* childStart represents the starting visual row
  1124.                           * for the child we're examining. */
  1125.     for (var i = 0; i < this.childData.length; ++i)
  1126.     {
  1127.         var child = this.childData[i];
  1128.         /* ignore hidden children */
  1129.         if (child.isHidden)
  1130.             continue;
  1131.         /* if this kid is the targetRow, we're done */
  1132.         if (childStart == targetRow)
  1133.             return (this._share.rowCache[targetRow] = child);
  1134.         /* if this kid contains the index, ask *it* to find the record */
  1135.         else if (targetRow <= childStart + child.visualFootprint) {
  1136.             /* this *has* to succeed */
  1137.             var rv = child.locateChildByVisualRow(targetRow, childStart + 1);
  1138.             //XXXASSERT (rv, "Can't find a row that *has* to be there.");
  1139.             /* don't cache this, the previous call to locateChildByVisualRow
  1140.              * just did. */
  1141.             return rv;
  1142.         }
  1143.  
  1144.         /* otherwise, get ready to ask the next kid */
  1145.         childStart += child.visualFootprint;
  1146.     }
  1147.  
  1148.     return null;
  1149. }
  1150.  
  1151. XTRootRecord.prototype.onVisualFootprintChanged =
  1152. function torr_vfpchange (start, amount)
  1153. {
  1154.     if (!this._treeView.frozen)
  1155.     {
  1156.         this.invalidateCache();
  1157.         this.visualFootprint += amount;
  1158.         if ("_treeView" in this && "tree" in this._treeView && 
  1159.             this._treeView.tree)
  1160.         {
  1161.             if (amount != 0)
  1162.                 this._treeView.tree.rowCountChanged (start, amount);
  1163.             else
  1164.                 this._treeView.tree.invalidateRow (start);
  1165.         }
  1166.     }
  1167.     else
  1168.     {
  1169.         if ("changeAmount"  in this._treeView)
  1170.             this._treeView.changeAmount += amount;
  1171.         else
  1172.             this._treeView.changeAmount = amount;
  1173.         if ("changeStart" in this._treeView)
  1174.             this._treeView.changeStart = 
  1175.                 Math.min (start, this._treeView.changeStart);
  1176.         else
  1177.             this._treeView.changeStart = start;
  1178.     }
  1179. }
  1180.  
  1181. /*
  1182.  * XULTreeView provides functionality of tree whose elements have multiple
  1183.  * levels of children.
  1184.  */
  1185.  
  1186. function XULTreeView(share)
  1187. {
  1188.     if (!share)
  1189.         share = new Object();
  1190.     this.childData = new XTRootRecord(this, share);
  1191.     this.childData.invalidateCache();
  1192.     this.tree = null;
  1193.     this.share = share;
  1194.     this.frozen = 0;
  1195. }
  1196.  
  1197. /* functions *you* should call to initialize and maintain the tree state */
  1198.  
  1199. /*
  1200.  * Changes to the tree contents will not cause the tree to be invalidated
  1201.  * until thaw() is called.  All changes will be pooled into a single invalidate
  1202.  * call.
  1203.  *
  1204.  * Freeze/thaws are nestable, the tree will not update until the number of
  1205.  * thaw() calls matches the number of freeze() calls.
  1206.  */
  1207. XULTreeView.prototype.freeze =
  1208. function xtv_freeze ()
  1209. {
  1210.     if (++this.frozen == 1)
  1211.     {
  1212.         this.changeStart = 0;
  1213.         this.changeAmount = 0;
  1214.     }
  1215.     //dd ("freeze " + this.frozen);
  1216. }
  1217.  
  1218. /*
  1219.  * Reflect any changes to the tee content since the last freeze.
  1220.  */
  1221. XULTreeView.prototype.thaw =
  1222. function xtv_thaw ()
  1223. {
  1224.     if (this.frozen == 0)
  1225.     {
  1226.         ASSERT (0, "not frozen");
  1227.         return;
  1228.     }
  1229.     
  1230.     if (--this.frozen == 0 && "changeStart" in this)
  1231.     {
  1232.         this.childData.onVisualFootprintChanged(this.changeStart,
  1233.                                                 this.changeAmount);
  1234.     }
  1235.     
  1236.     if ("needsResort" in this) {
  1237.         this.childData.resort();
  1238.         delete this.needsResort;
  1239.     }
  1240.     
  1241.     
  1242.     delete this.changeStart;
  1243.     delete this.changeAmount;
  1244. }
  1245.  
  1246. XULTreeView.prototype.saveBranchState =
  1247. function xtv_savebranch (target, source, recurse)
  1248. {
  1249.     var len = source.length;
  1250.     for (var i = 0; i < len; ++i)
  1251.     {
  1252.         if (source[i].isContainerOpen)
  1253.         {
  1254.             target[i] = new Object();
  1255.             target[i].name = source[i]._colValues["col-0"];
  1256.             if (recurse)
  1257.                 this.saveBranchState (target[i], source[i].childData, true);
  1258.         }
  1259.     }
  1260. }
  1261.  
  1262. XULTreeView.prototype.restoreBranchState =
  1263. function xtv_restorebranch (target, source, recurse)
  1264. {
  1265.     for (var i in source)
  1266.     {
  1267.         if (typeof source[i] == "object")
  1268.         {
  1269.             var name = source[i].name;
  1270.             var len = target.length;
  1271.             for (var j = 0; j < len; ++j)
  1272.             {
  1273.                 if (target[j]._colValues["col-0"] == name &&
  1274.                     "childData" in target[j])
  1275.                 {
  1276.                     //dd ("opening " + name);
  1277.                     target[j].open();
  1278.                     if (recurse)
  1279.                     {
  1280.                         this.restoreBranchState (target[j].childData,
  1281.                                                  source[i], true);
  1282.                     }
  1283.                     break;
  1284.                 }
  1285.             }
  1286.         }
  1287.     }
  1288. }
  1289.  
  1290. /* scroll the line specified by |line| to the center of the tree */
  1291. XULTreeView.prototype.centerLine =
  1292. function xtv_ctrln (line)
  1293. {
  1294.     var first = this.tree.getFirstVisibleRow();
  1295.     var last = this.tree.getLastVisibleRow();
  1296.     this.scrollToRow(line - total / 2);
  1297. }
  1298.  
  1299. /*
  1300.  * functions the tree will call to retrieve the list state (nsITreeView.)
  1301.  */
  1302.  
  1303. XULTreeView.prototype.__defineGetter__("rowCount", xtv_getRowCount);
  1304. function xtv_getRowCount ()
  1305. {
  1306.     if (!this.childData)
  1307.       return 0;
  1308.  
  1309.     return this.childData.visualFootprint;
  1310. }
  1311.  
  1312. XULTreeView.prototype.isContainer =
  1313. function xtv_isctr (index)
  1314. {
  1315.     var row = this.childData.locateChildByVisualRow (index);
  1316.     /*
  1317.     ASSERT(row, "bogus row");
  1318.     var rv = Boolean(row && row.childData);
  1319.     dd ("isContainer: row " + index + " returning " + rv);
  1320.     return rv;
  1321.     */
  1322.  
  1323.     return Boolean(row && ("alwaysHasChildren" in row || "childData" in row));
  1324. }
  1325.  
  1326. XULTreeView.prototype.__defineGetter__("selectedIndex", xtv_getsel);
  1327. function xtv_getsel()
  1328. {
  1329.     if (!this.tree || this.tree.view.selection.getRangeCount() < 1)
  1330.         return -1;
  1331.  
  1332.     var min = new Object();
  1333.     this.tree.view.selection.getRangeAt(0, min, {});
  1334.     return min.value;
  1335. }
  1336.  
  1337. XULTreeView.prototype.__defineSetter__("selectedIndex", xtv_setsel);
  1338. function xtv_setsel(i)
  1339. {
  1340.     this.tree.view.selection.clearSelection();
  1341.     if (i != -1)
  1342.         this.tree.view.selection.timedSelect (i, 500);
  1343.     return i;
  1344. }
  1345.  
  1346. XULTreeView.prototype.scrollTo = BasicOView.prototype.scrollTo;
  1347.  
  1348. XULTreeView.prototype.isContainerOpen =
  1349. function xtv_isctropen (index)
  1350. {
  1351.     var row = this.childData.locateChildByVisualRow (index);
  1352.     /*
  1353.     ASSERT(row, "bogus row");
  1354.     var rv = Boolean(row && row.isContainerOpen);
  1355.     dd ("isContainerOpen: row " + index + " returning " + rv);
  1356.     return rv;
  1357.     */
  1358.     return row && row.isContainerOpen;
  1359. }
  1360.  
  1361. XULTreeView.prototype.toggleOpenState =
  1362. function xtv_toggleopen (index)
  1363. {
  1364.     var row = this.childData.locateChildByVisualRow (index);
  1365.     //ASSERT(row, "bogus row");
  1366.     if (row)
  1367.     {
  1368.         if (row.isContainerOpen)
  1369.             row.close();
  1370.         else
  1371.             row.open();
  1372.     }
  1373. }
  1374.  
  1375. XULTreeView.prototype.isContainerEmpty =
  1376. function xtv_isctrempt (index)
  1377. {
  1378.     var row = this.childData.locateChildByVisualRow (index);
  1379.     /*
  1380.     ASSERT(row, "bogus row");
  1381.     var rv = Boolean(row && (row.childData.length == 0));
  1382.     dd ("isContainerEmpty: row " + index + " returning " + rv);
  1383.     return rv;
  1384.     */
  1385.     if ("alwaysHasChildren" in row)
  1386.         return false;
  1387.  
  1388.     if (!row || !("childData" in row))
  1389.         return true;
  1390.  
  1391.     return !row.childData.length;
  1392. }
  1393.  
  1394. XULTreeView.prototype.isSeparator =
  1395. function xtv_isseparator (index)
  1396. {
  1397.     return false;
  1398. }
  1399.  
  1400. XULTreeView.prototype.getParentIndex =
  1401. function xtv_getpi (index)
  1402. {
  1403.     if (index < 0)
  1404.         return -1;
  1405.     
  1406.     var row = this.childData.locateChildByVisualRow (index);
  1407.     //if (!ASSERT(row, "bogus row " + index))
  1408.     //    return -1;
  1409.     
  1410.     var rv = row.parentRecord.calculateVisualRow();
  1411.     //dd ("getParentIndex: row " + index + " returning " + rv);
  1412.     return (rv != null) ? rv : -1;
  1413. }
  1414.  
  1415. XULTreeView.prototype.hasNextSibling =
  1416. function xtv_hasnxtsib (rowIndex, afterIndex)
  1417. {
  1418.     var row = this.childData.locateChildByVisualRow (rowIndex);
  1419.     /*
  1420.     ASSERT(row, "bogus row");
  1421.     rv = Boolean(row.childIndex < row.parentRecord.childData.length - 1);
  1422.     dd ("hasNextSibling: row " + rowIndex + ", after " + afterIndex +
  1423.         " returning " + rv);
  1424.     return rv;
  1425.     */
  1426.     return row.childIndex < row.parentRecord.childData.length - 1;
  1427. }
  1428.  
  1429. XULTreeView.prototype.getLevel =
  1430. function xtv_getlvl (index)
  1431. {
  1432.     var row = this.childData.locateChildByVisualRow (index);
  1433.     /*
  1434.     ASSERT(row, "bogus row");
  1435.     var rv = row.level;
  1436.     dd ("getLevel: row " + index + " returning " + rv);
  1437.     return rv;
  1438.     */
  1439.     if (!row)
  1440.         return 0;
  1441.     
  1442.     return row.level;
  1443. }
  1444.  
  1445. XULTreeView.prototype.getImageSrc =
  1446. function xtv_getimgsrc (index, col)
  1447. {
  1448. }
  1449.  
  1450. XULTreeView.prototype.getProgressMode =
  1451. function xtv_getprgmode (index, col)
  1452. {
  1453. }
  1454.  
  1455. XULTreeView.prototype.getCellValue =
  1456. function xtv_getcellval (index, col)
  1457. {
  1458. }
  1459.  
  1460. XULTreeView.prototype.getCellText =
  1461. function xtv_getcelltxt (index, col)
  1462. {
  1463.     var row = this.childData.locateChildByVisualRow (index);
  1464.     //ASSERT(row, "bogus row " + index);
  1465.  
  1466.     var ary = col.id.match (/:(.*)/);
  1467.     if (ary)
  1468.         col = ary[1];
  1469.  
  1470.     if (row && row._colValues && col in row._colValues)
  1471.         return row._colValues[col];
  1472.     else
  1473.         return "";
  1474. }
  1475.  
  1476. XULTreeView.prototype.getCellProperties =
  1477. function xtv_cellprops (row, col, properties)
  1478. {}
  1479.  
  1480. XULTreeView.prototype.getColumnProperties =
  1481. function xtv_colprops (col, properties)
  1482. {}
  1483.  
  1484. XULTreeView.prototype.getRowProperties =
  1485. function xtv_rowprops (index, properties)
  1486. {}
  1487.  
  1488. XULTreeView.prototype.isSorted =
  1489. function xtv_issorted (index)
  1490. {
  1491.     return false;
  1492. }
  1493.  
  1494. XULTreeView.prototype.canDrop =
  1495. function xtv_drop (index, orientation)
  1496. {
  1497.     var row = this.childData.locateChildByVisualRow (index);
  1498.     //ASSERT(row, "bogus row " + index);
  1499.     return (row && ("canDrop" in row) && row.canDrop(orientation));
  1500. }
  1501.  
  1502. XULTreeView.prototype.drop =
  1503. function xtv_drop (index, orientation)
  1504. {
  1505.     var row = this.childData.locateChildByVisualRow (index);
  1506.     //ASSERT(row, "bogus row " + index);
  1507.     return (row && ("drop" in row) && row.drop(orientation));
  1508. }
  1509.  
  1510. XULTreeView.prototype.setTree =
  1511. function xtv_seto (tree)
  1512. {
  1513.     this.childData.invalidateCache();
  1514.     this.tree = tree;
  1515. }
  1516.  
  1517. XULTreeView.prototype.cycleHeader =
  1518. function xtv_cyclehdr (col)
  1519. {
  1520. }
  1521.  
  1522. XULTreeView.prototype.selectionChanged =
  1523. function xtv_selchg ()
  1524. {
  1525. }
  1526.  
  1527. XULTreeView.prototype.cycleCell =
  1528. function xtv_cyclecell (row, col)
  1529. {
  1530. }
  1531.  
  1532. XULTreeView.prototype.isEditable =
  1533. function xtv_isedit (row, col)
  1534. {
  1535.     return false;
  1536. }
  1537.  
  1538. XULTreeView.prototype.setCellValue =
  1539. function xtv_setct (row, col, value)
  1540. {
  1541. }
  1542.  
  1543. XULTreeView.prototype.setCellText =
  1544. function xtv_setct (row, col, value)
  1545. {
  1546. }
  1547.  
  1548. XULTreeView.prototype.performAction =
  1549. function xtv_pact (action)
  1550. {
  1551. }
  1552.  
  1553. XULTreeView.prototype.performActionOnRow =
  1554. function xtv_pactrow (action)
  1555. {
  1556. }
  1557.  
  1558. XULTreeView.prototype.performActionOnCell =
  1559. function xtv_pactcell (action)
  1560. {
  1561. }
  1562.  
  1563. XULTreeView.prototype.onRouteFocus =
  1564. function xtv_rfocus (event)
  1565. {
  1566.     if ("onFocus" in this)
  1567.         this.onFocus(event);
  1568. }
  1569.  
  1570. XULTreeView.prototype.onRouteBlur =
  1571. function xtv_rblur (event)
  1572. {
  1573.     if ("onBlur" in this)
  1574.         this.onBlur(event);
  1575. }
  1576.  
  1577. XULTreeView.prototype.onRouteDblClick =
  1578. function xtv_rdblclick (event)
  1579. {
  1580.     if (!("onRowCommand" in this) || event.target.localName != "treechildren")
  1581.         return;
  1582.  
  1583.     var rowIndex = this.tree.view.selection.currentIndex;
  1584.     if (rowIndex == -1 || rowIndex > this.rowCount)
  1585.         return;
  1586.     var rec = this.childData.locateChildByVisualRow(rowIndex);
  1587.     if (!rec)
  1588.     {
  1589.         ASSERT (0, "bogus row index " + rowIndex);
  1590.         return;
  1591.     }
  1592.  
  1593.     this.onRowCommand(rec, event);
  1594. }
  1595.  
  1596. XULTreeView.prototype.onRouteKeyPress =
  1597. function xtv_rkeypress (event)
  1598. {
  1599.     var rec;
  1600.     var rowIndex;
  1601.     
  1602.     if ("onRowCommand" in this && (event.keyCode == 13 || event.charCode == 32))
  1603.     {
  1604.         if (!this.selection)
  1605.             return;
  1606.         
  1607.         rowIndex = this.tree.view.selection.currentIndex;
  1608.         if (rowIndex == -1 || rowIndex > this.rowCount)
  1609.             return;
  1610.         rec = this.childData.locateChildByVisualRow(rowIndex);
  1611.         if (!rec)
  1612.         {
  1613.             ASSERT (0, "bogus row index " + rowIndex);
  1614.             return;
  1615.         }
  1616.  
  1617.         this.onRowCommand(rec, event);
  1618.     }
  1619.     else if ("onKeyPress" in this)
  1620.     {
  1621.         rowIndex = this.tree.view.selection.currentIndex;
  1622.         if (rowIndex != -1 && rowIndex < this.rowCount)
  1623.         {
  1624.             rec = this.childData.locateChildByVisualRow(rowIndex);
  1625.             if (!rec)
  1626.             {
  1627.                 ASSERT (0, "bogus row index " + rowIndex);
  1628.                 return;
  1629.             }
  1630.         }
  1631.         else
  1632.         {
  1633.             rec = null;
  1634.         }
  1635.         
  1636.         this.onKeyPress(rec, event);
  1637.     }
  1638. }
  1639.  
  1640. /******************************************************************************/
  1641.  
  1642. function xtv_formatRecord (rec, indent)
  1643. {
  1644.     var str = "";
  1645.     
  1646.     for (var i in rec._colValues)
  1647.             str += rec._colValues[i] + ", ";
  1648.     
  1649.     str += "[";
  1650.     
  1651.     str += rec.calculateVisualRow() + ", ";
  1652.     str += rec.childIndex + ", ";
  1653.     str += rec.level + ", ";
  1654.     str += rec.visualFootprint + ", ";
  1655.     str += rec.isHidden + "]";
  1656.     
  1657.     return (indent + str);
  1658. }
  1659.     
  1660. function xtv_formatBranch (rec, indent, recurse)
  1661. {
  1662.     var str = "";
  1663.     for (var i = 0; i < rec.childData.length; ++i)
  1664.     {
  1665.         str += xtv_formatRecord (rec.childData[i], indent) + "\n";
  1666.         if (recurse)
  1667.         {
  1668.             if ("childData" in rec.childData[i])
  1669.                 str += xtv_formatBranch(rec.childData[i], indent + "  ",
  1670.                                         --recurse);
  1671.         }
  1672.     }
  1673.  
  1674.     return str;
  1675. }
  1676.     
  1677.